//package com.dz.config;
//
//import com.dz.bean.user;
//import org.apache.shiro.cache.Cache;
//import org.apache.shiro.cache.CacheManager;
//import org.apache.shiro.session.Session;
//import org.apache.shiro.session.mgt.SessionManager;
//import org.apache.shiro.subject.Subject;
//import org.apache.shiro.web.filter.AccessControlFilter;
//import org.apache.shiro.web.util.WebUtils;
//import org.mybatis.logging.Logger;
//import org.mybatis.logging.LoggerFactory;
//
//import javax.servlet.ServletRequest;
//import javax.servlet.ServletResponse;
//import java.io.Serializable;
//import java.util.ArrayDeque;
//import java.util.Deque;
//
//public class KickoutSessionFilter extends AccessControlFilter {
//    private static final Logger logger = LoggerFactory
//            .getLogger(KickoutSessionFilter.class);
//
//    private String kickoutUrl; // 踢出后到的地址
//    private boolean kickoutAfter = false; // 踢出之前登录的/之后登录的用户 默认false踢出之前登录的用户
//    private int maxSession = 1; // 同一个帐号最大会话数 默认1
//    private SessionManager sessionManager;
//    private Cache<String, Deque<Serializable>> cache;
//
//    public void setKickoutUrl(String kickoutUrl) {
//        this.kickoutUrl = kickoutUrl;
//    }
//
//    public void setKickoutAfter(boolean kickoutAfter) {
//        this.kickoutAfter = kickoutAfter;
//    }
//
//    public void setMaxSession(int maxSession) {
//        this.maxSession = maxSession;
//    }
//
//    public void setSessionManager(SessionManager sessionManager) {
//        this.sessionManager = sessionManager;
//    }
//
//    // 设置Cache的key的前缀
//    public void setCacheManager(CacheManager cacheManager) {
//        //必须和ehcache缓存配置中的缓存name一致
//        this.cache = cacheManager.getCache("shiro-activeSessionCache");
//    }
//
//
//    //    isAccessAllowed：表示是否允许访问；mappedValue就是[urls]配置中拦截器参数部分，如果允许访问返回true，否则false；
////    onAccessDenied：表示当访问拒绝时是否已经处理了；如果返回true表示需要继续处理；如果返回false表示该拦截器实例已经处理了，将直接返回即可。
//    @Override
//    protected boolean isAccessAllowed(ServletRequest servletRequest, ServletResponse servletResponse, Object o) throws Exception {
//        return false;
//    }
//
//    @Override
//    protected boolean onAccessDenied(ServletRequest request, ServletResponse response) throws Exception {
//        Subject subject = getSubject(request, response);
//        // 没有登录授权 且没有记住我
//        if (!subject.isAuthenticated() && !subject.isRemembered()) {
//            // 如果没有登录，直接进行之后的流程
//            return true;
//        }
//        Session session = subject.getSession();
//        logger.debug("==session时间设置：" + String.valueOf(session.getTimeout())
//                + "===========");
//        try {
//            // 当前用户
//            user user = (user) subject.getPrincipal();
//            String username = user.getUsername();
//            logger.debug("===当前用户username:==" + username);
//            Serializable sessionId = session.getId();
//            logger.debug("===当前用户sessionId：==" + sessionId);
//            // 读取缓存用户 没有就存入
//            Deque<Serializable> deque = cache.get(username);
//            logger.debug("===当前deque：==" + deque);
//            if (deque == null) {
//                // 初始化队列
//                deque = new ArrayDeque<Serializable>();
//            }
//            // 如果队列里没有此sessionId，且用户没有被踢出；放入队列
//            if (!deque.contains(sessionId)
//                    && session.getAttribute("kickout") == null) {
//                // 将sessionId存入队列
//                deque.push(sessionId);
//                // 将用户的sessionId队列缓存
//                cache.put(username, deque);
//            }
//            // 如果队列里的sessionId数超出最大会话数，开始踢人
//            while (deque.size() > maxSession) {
//                logger.debug("===deque队列长度：==" + deque.size());
//                Serializable kickoutSessionId = null;
//                // 是否踢出后来登录的，默认是false；即后者登录的用户踢出前者登录的用户；
//                if (kickoutAfter) { // 如果踢出后者
//                    kickoutSessionId = deque.removeFirst();
//                } else { // 否则踢出前者
//                    kickoutSessionId = deque.removeLast();
//                }
//                // 踢出后再更新下缓存队列
//                cache.put(username, deque);
//                try {
//                    // 获取被踢出的sessionId的session对象
//                    Session kickoutSession = sessionManager
//                            .getSession(new DefaultSessionKey(kickoutSessionId));
//                    if (kickoutSession != null) {
//                        // 设置会话的kickout属性表示踢出了
//                        kickoutSession.setAttribute("kickout", true);
//                    }
//                } catch (Exception e) {// ignore exception
//                }
//            }
//            // ajax请求
//
//            // 如果被踢出了，(前者或后者)直接退出，重定向到踢出后的地址
//            if ((Boolean) session.getAttribute("kickout") != null
//                    && (Boolean) session.getAttribute("kickout") == true) {
//                // 会话被踢出了
//                try {
//                    // 退出登录
//                    subject.logout();
//                } catch (Exception e) { // ignore
//                }
//                saveRequest(request);
//                logger.debug("==踢出后用户重定向的路径kickoutUrl:" + kickoutUrl);
//                // 重定向
//                WebUtils.issueRedirect(request, response, kickoutUrl);
//                return false;
//            }
//            return true;
//        } catch (Exception e) { // ignore
//            //重定向到登录界面
//            WebUtils.issueRedirect(request, response, "/login");
//            return false;
//        }
//    }
//}
